About

This document contains analysis of recent data regarding wage taxes, licensing, zoning, and mobility related to Philadelphia’s nighttime economy.

Analysis by Michael Fichman of Philadelphia City Council’s Arts and Culture Task Force, Nightlife Committee.

Wage tax revenue data obtained from Philadelphia Controller’s Office, updated for the 2021 tax year, 3/8/2022. All other data obtained from Open Data Philly or Indigo Bike Share, April, 2022.

For code base used to synthesize these data, refer to the code download button at the top of this document or visit the github repository for this page - https://github.com/mafichman/ACTF_nightlife

Questions, contact

1. Hospitality Wage Tax Receipts, 2014-2022

Quarterly wage tax receipts for industries in the arts and hospitality were dramatically impacted by the pandemic which began in March, 2020, and have not fully recovered.

Data on nighttime-specific industries tend not to be subdivided by day and nighttime activity in wage tax data.

All dollar values adjusted for inflation (2021 dollars) based on BLS guidelines.

1.1. Tax revenues by year in millions of dollars

year Arts, Entertainment, and Other Recreation Hotels Restaurants
2014 8.78 9.47 36.89
2015 14.35 12.10 52.20
2016 16.06 12.18 55.62
2017 16.21 11.87 57.90
2018 16.65 12.44 59.19
2019 18.48 12.83 61.75
2020 13.26 6.53 41.47
2021 12.63 7.81 43.26

1.3 Tax Revenue in Pandemic Year 1

Year over year losses - Pandemic Year 1 - Wage tax revenues during March, 2020 - December, 2021 relative to March, 2019 - February, 2020.

As gross (in millions of US dollars, adjust for inflation to 2021 dollars) and as a percentage change.

variable Pre-pandemic year Pandemic year 1 Pct_Change
Arts_Entertainment_Recreation 16.51 10.81 -34.52
Hotels 11.55 4.75 -58.87
Restaurants 55.84 33.08 -40.76
Total 1964.94 1787.38 -9.04

1.3 Tax Loss Estimates and Recovery Timeline

Tax revenues by the end of 2021 are still falling short of expected, meaning that recovery has not been achieved. Expected revenues are based on 2014-2020(TY Q1) trends (based on year, quarter and sector) and adjusted for inflation to 2021 dollars.

Projected revenues are climbing back towards pre-pandemic status-quo.

Wage tax losses since the beginning of 2020, relative to projections based on 2014-19, are still as much as $2 million per quarter by sector.

Total tax shortfalls (compared to pre-pandemic trend) since March, 2020, in millions (2021 dollars)

sector Est Loss (millions)
Arts, Entertainment, and Other Recreation -14.91
Hotels -18.69
Restaurants -36.96

2. Save Our Stages Grants

This repo contains interactive maps and analysis of SBA Grants given as part of the “Save Our Stages” Act passed by the US Congress.

The data being used here have been geocoded using the Google Geocoding service, which is highly accurate but not error free. Data are current as of August 17, 2021, and are available from the SBA.

2.2. Grantee map

The following map shows the grantee locations.

2.2. Philadelphia SVOG grant breakdown:

Num Grants Grants Total Mean Grant Median Grant
68 88135486 1296110 247143.4

2.3. Phila SVOG searchable database:

3. Zoning and Assembly Licenses

A Special Assembly Occupancy License (SAOL) is the highest form of license for assembly for entertainment for 50+ people (non-seated). In combination with an amusement license and regular business licenses, this is what you usually need to open a venue. These licenses are difficult to get, and can be expensive. Much expense and risk is incurred when you need a zoning variance (and council approval) or to go through a zoning process triggered by a special exception.

SAOLs are only available by right in very few zoning categories, most of these the most expensive land in Center City. Recently, people have NOT been taking out licenses in CC, they have been going through zoning processes in nearby neighborhoods with transit access and clusters of amenities, and SAOLs have been going extinct much faster than they are being created.

There are a few opportunities for Council to make SAOLs easier to access - one is by supporting operators in getting variances, two is by opening up more zoning districts to SAOLs by right.

24HrPHL put together a “playbook” for how to obtain a SAOL in 2019 that was vetted by then L&I Commissioner Dave Perri.

The number of active SAOLs in Philadelphia is declining, and the places where people want to (or can afford to) build creative spaces are not the places where they are available by-right. Only 28% of active SAOLs are in by-right districts.

Frequently, people are applying for SAOLs in CMX-2.5 properties - where this use is availabe by special exception Opening CMX-2.5 to by-right SAOL would more than double the amount of commercial parcels eligible for by-right SAOL.

## Reading layer `Zoning_BaseDistricts' from data source 
##   `https://opendata.arcgis.com/datasets/0bdb0b5f13774c03abf8dc2f1aa01693_0.geojson' 
##   using driver `GeoJSON'
## Simple feature collection with 29151 features and 13 fields
## Geometry type: MULTIPOLYGON
## Dimension:     XY
## Bounding box:  xmin: -75.28023 ymin: 39.87082 xmax: -74.95581 ymax: 40.13786
## Geodetic CRS:  WGS 84
## Reading layer `OGRGeoJSON' from data source 
##   `https://phl.carto.com/api/v2/sql?q=SELECT+*+FROM+business_licenses&filename=business_licenses&format=geojson&skipfields=cartodb_id' 
##   using driver `GeoJSON'
## Simple feature collection with 0 features and 0 fields
## Bounding box:  xmin: NA ymin: NA xmax: NA ymax: NA
## Geodetic CRS:  WGS 84

3.1. Summary of SAOL-eligible Districts

Assembly is allowed by right in four zoning districts, and by special exception in four others.

Here are some very basic descriptions of these types of zones, with information sourced from Anastasio Law - http://phillyzoning.com/

By Right

-CA-2 - Auto-oriented commercial

-CMX-4 - Commercial Mixed Use, mostly found in Center City or along major arterials like Broad, Market or Chestnut Streets

-CMX-5 - Commercial Mixed Use, found in the core of Center City office districts

-ICMX - Commercial/Industrial uses designed as a buffer between commercial and residential uses

Special Exception

-CMX-2 - Neighborhood commercial corridor, mixed use - e.g. Baltimore Avenue, Germantown Avenue

-CMX-2.5 - Commercial mixed use designed to promote pedestrian-friendly uses

-CMX-3 - Lower density commercial mixed use than CMX 4&5, found on major corridors like Kensington Ave, South St, Broad south of Washington.

-IRMX - “Low impact” industrial, including artist spaces

City-wide, roughly 26% of commercial districts (by area) are zoned for assembly by right. Much of this area is in large Center City tracts where creative spaces are unlikely to be developed due to cost or constraints of the land.

assembly_allowed sum_area Pct
By-Right 78538211 [US_survey_foot^2] 26.37 [1]
Not Permitted 42801765 [US_survey_foot^2] 14.37 [1]
Special Exception 176471533 [US_survey_foot^2] 59.26 [1]

3.2. City-wide zoning examination

The majority of areas zoned for assembly by-right are in Center City. This is not where most development of creative entertainment is taking place.

Take a look at districts outside Center City like Fishtown - by-right SAOL is not really allowed by-right anywhere.

When the license data from OpenDataPhilly are back online - this map will contain SAOL licence locations as well.

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

3.4. SAOL issuance timeline

Very few SAOLs are issued in Philly, and especially few the last two years. The following chart shows the date when SAOL’s were issued. 2006 probably has data inflated by the start of the record-keeping period for these data.

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

Record numbers of licenses became inactive in 2020… The following chart shows the years in which SAOL’s became inactive. Meanwhile, the real estate market is getting much more competitive and entertainment wages are slow to recover (see Section 1 of this document).

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

Among active Special Assembly Occupancy Licenses, only 35 (28%) were permitted by right in the zones where they were created.

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

Next steps -

How has by-right vs. exception/variance changed over time? How is this related to property costs / tax rates / valuation? How many SAOLs are also getting amusement licenses? How many can we verify are music venues? Are temporary assembly licenses in the data set?

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

Next up for analysis

overlays, Mobility trends, PPP data, Illuminate the Arts grants

Bike Share and Mobility

Philadelphia Bike Share Q2, 2021

ggplot(dat2 %>%
         group_by(week) %>%
         tally())+
  geom_line(aes(x = week, y = n))+
  labs(title="Bike share trips per week Philadelphia, 2021",
       x="Week (2021)", 
       y="Trips")+
  plotTheme

ggplot(dat2 %>% mutate(hour = hour(mdy_hm(start_time)),
                       month = month(mdy_hm(start_time)),
                       day_type = ifelse(dotw %in% c("Sat", "Sun", "Fri"), "Weekend", "Weekday"),
                       month = case_when(month == 1 ~ "01-2021",
                                         month == 2 ~ "02-2021",
                                         month == 3 ~ "03-2021",
                                         month == 4 ~ "04-2021",
                                         month == 5 ~ "05-2021",
                                         month == 6 ~ "06-2021",
                                         month == 7 ~ "07-2021",
                                         month == 8 ~ "08-2021",
                                         month == 9 ~ "09-2021",
                                         month == 10 ~ "10-2021",
                                         month == 11 ~ "11-2021",
                                         month == 12 ~ "12-2021")))+
  geom_rect(aes(xmin=0,
                xmax = 6,
                ymin = 0,
                ymax = Inf), fill = 'light grey', alpha = 0.05)+
  geom_rect(aes(xmin=18,
                xmax = 24,
                ymin = 0,
                ymax = Inf), fill = 'light grey', alpha = 0.05)+
  geom_freqpoly(aes(hour, color = day_type), binwidth = 1)+
  scale_x_continuous(name="Hour", breaks=seq(0,24,6))+
  labs(title="Total Bike Share Trips",
       subtitle = "Philadelphia, 2021",
       x="Hour", 
       y="Trips")+
  facet_wrap(~month)+
  theme(legend.title = element_blank())+
  plotTheme
---
title: "ACTF Nightlife Data Analysis and Outputs"
author: "Michael Fichman"
date: "May 8, 2022"
output: 
  html_document:
    toc: true
    toc_float: true
    code_folding: "hide"
    code_download: true
---

```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE)

library(tidyverse)
library(readxl)
library(lubridate)
library(kableExtra)
library(leaflet)
library(leaflet.extras)
library(leaflet.providers)
library(sf)
library(ggmap)
library(DT)
library(tigris)
library(mapview)

plotTheme <- theme(
  plot.title =element_text(size=12),
  plot.subtitle = element_text(size=8),
  plot.caption = element_text(size = 6),
  axis.text.x = element_text(size = 10, angle = 45, hjust = 1),
  axis.text.y = element_text(size = 10),
  axis.title.y = element_text(size = 10),
  # Set the entire chart region to blank
  panel.background=element_blank(),
  plot.background=element_blank(),
  #panel.border=element_rect(colour="#F0F0F0"),
  # Format the grid
  panel.grid.major=element_line(colour="#D0D0D0",size=.75),
  axis.ticks=element_blank())

mapTheme <- theme(plot.title =element_text(size=12),
                  plot.subtitle = element_text(size=8),
                  plot.caption = element_text(size = 6),
                  axis.line=element_blank(),
                  axis.text.x=element_blank(),
                  axis.text.y=element_blank(),
                  axis.ticks=element_blank(),
                  axis.title.x=element_blank(),
                  axis.title.y=element_blank(),
                  panel.background=element_blank(),
                  panel.border=element_blank(),
                  panel.grid.major=element_line(colour = 'transparent'),
                  panel.grid.minor=element_blank(),
                  legend.direction = "vertical", 
                  legend.position = "right",
                  plot.margin = margin(1, 1, 1, 1, 'cm'),
                  legend.key.height = unit(1, "cm"), legend.key.width = unit(0.2, "cm"))



wage_tax_revenue <- read_excel("~/GitHub/ACTF_nightlife/data/wage-tax-revenue.xlsx") %>%
  rename(Arts_Entertainment_Recreation = 'Arts, Entertainment, and Other Recreation') %>%
  mutate(year = year(ymd(date)),
         month = month(ymd(date)))

# Load data, remove duplicates, adjust for inflation to 2021 dollars
wage_tax_revenue_new <- read.csv("https://phl-budget-datasette.herokuapp.com/revenue/wage-collections-by-sector.csv?sector__in=%5B%22Hotels%22%2C+%22Arts%2C+Entertainment%2C+and+Other+Recreation%22%2C+%22Restaurants%22%5D&_sort_desc=date&_size=max") %>%
    mutate(year = year(ymd(date)),
           month = month(ymd(date))) %>%
  select(-rowid) %>%
  group_by(fiscal_year, fiscal_quarter, sector) %>%
  slice(1) %>%
  ungroup() %>%
  mutate(total = case_when(year == 2014 ~ total * 1.12,
                           year == 2015 ~ total * 1.12,
                           year == 2016 ~ total * 1.10,
                           year == 2017 ~ total * 1.08,
                           year == 2018 ~ total * 1.06,
                           year == 2019 ~ total * 1.04,
                           year == 2020 ~ total * 1.01,
                           year == 2021 ~ total))
```

# About

This document contains analysis of recent data regarding wage taxes, licensing, zoning, and mobility related to Philadelphia's nighttime economy.

Analysis by Michael Fichman of Philadelphia City Council's Arts and Culture Task Force, Nightlife Committee. 

Wage tax revenue data obtained from Philadelphia Controller's Office, updated for the 2021 tax year, 3/8/2022. All other data obtained from Open Data Philly or Indigo Bike Share, April, 2022.

For code base used to synthesize these data, refer to the code download button at the top of this document or visit the github repository for this page - https://github.com/mafichman/ACTF_nightlife

Questions, contact mfichman@upenn.edu

# 1. Hospitality Wage Tax Receipts, 2014-2022

Quarterly wage tax receipts for industries in the arts and hospitality were dramatically impacted by the pandemic which began in March, 2020, and have not fully recovered. 

Data on nighttime-specific industries tend not to be subdivided by day and nighttime activity in wage tax data.

All dollar values adjusted for inflation (2021 dollars) based on BLS guidelines.

```{r message=FALSE, warning=FALSE, echo=FALSE}
wage_tax_revenue_new %>%
  group_by(date, sector) %>%
  summarize(sum = sum(total),
            sum = round((sum/1000000), digits = 2)) %>%
  ggplot()+
  geom_line(aes(x = as.Date(date), y=sum, color = sector))+
  ylab("Quarterly Wage Tax Revenue ($ Millions)")+
  xlab("Date")+
  labs(
    title = "Philadelphia Wage Tax Receipts for Three Hospitality Industries",
    subtitle = "Adjusted for inflation to 2021 Dollars. Source: Philadelphia City Controller's Office, BLS")+
  plotTheme
```

## 1.1. Tax revenues by year in millions of dollars

```{r message=FALSE, warning=FALSE, echo=FALSE}
wage_tax_revenue_new %>%
  group_by(year, sector) %>%
  summarize(sum = sum(total)) %>%
  mutate(sum = round((sum/1000000), digits = 2)) %>%
  spread(sector, sum) %>%
  kable() %>%
  kable_styling()

```
## 1.3 Tax Revenue in Pandemic Year 1

Year over year losses - Pandemic Year 1 - Wage tax revenues during March, 2020 - December, 2021 relative to March, 2019 - February, 2020.

As gross (in millions of US dollars, adjust for inflation to 2021 dollars) and as a percentage change.

```{r message=FALSE, warning=FALSE, echo=FALSE}
wage_tax_revenue %>%
  mutate(pandemic = case_when(year == 2020 & month >=3 ~ "March2020_Feb2021",
                              year == 2021 ~ "March2020_Feb2021",
                              year == 2020 & month <3 ~ "March2019_Feb2020",
                              year == 2019 & month > 3 ~ "March2019_Feb2020")) %>%
  filter(is.na(pandemic) == FALSE) %>%
  select(-date, -month, -year ) %>%
  gather(-pandemic, key = "variable", value = "value") %>%
  group_by(pandemic, variable) %>%
  summarize(sum = sum(value)) %>%
  mutate(sum = round((sum/1000000), digits = 2)) %>%
  spread(pandemic, sum) %>%
  mutate(Pct_Change = round(100*((March2020_Feb2021 - March2019_Feb2020)/ March2019_Feb2020), digits = 2)) %>%
  rename("Pre-pandemic year" = March2019_Feb2020,
         "Pandemic year 1" = March2020_Feb2021) %>%
  kable() %>%
  kable_styling()

```

## 1.3 Tax Loss Estimates and Recovery Timeline

Tax revenues by the end of 2021 are still falling short of expected, meaning that recovery has not been achieved. Expected revenues are based on 2014-2020(TY Q1) trends (based on year, quarter and sector) and adjusted for inflation to 2021 dollars.

```{r model, warning = FALSE, message = FALSE, include=FALSE}
model <- lm(data = wage_tax_revenue_new %>% filter(year < 2020) %>% select(sector, fiscal_quarter, year, total), total ~ fiscal_quarter + year + sector)

summary(model)


predict(model, wage_tax_revenue_new) %>% 
  as.data.frame() %>% 
  rename(pred = ".") %>% 
  cbind(., wage_tax_revenue_new) -> wage_tax_revenue_new
```

Projected revenues are climbing back towards pre-pandemic status-quo.

```{r projection_chart, warning = FALSE, message = FALSE, echo=FALSE}
wage_tax_revenue_new %>%
  group_by(date, sector, year, month) %>% 
  summarize(Actual = sum(total), 
            Pre.Pandemic.Trend = sum(pred)) %>% 
  filter(year >= 2018) %>%
  ungroup() %>%
  select(date, sector, Actual, Pre.Pandemic.Trend) %>%
  gather(-sector, - date, key = "variable", value = "value") %>%
  mutate(value_m = round(value/1000000, digits = 2)) %>%
  ggplot()+
    geom_line(aes(x = as.Date(date), y=value_m, color = variable))+ 
    geom_line(aes(x = as.Date(date), y=value_m, color = variable), 
            linetype = "dashed", alpha = 0.5)+
  facet_wrap(~sector)+
    ylab("Quarterly Wage Tax Revenue ($ Millions)")+
    xlab("Date")+
    labs(
        title = "Philadelphia Wage Tax Shortfalls",
        subtitle = "Adjusted for inflation to 2021 Dollars.\n Trend based on 2014-2019 regression controlling for year, quarter, sector. \nSource: Philadelphia City Controller's Office, BLS")+
    plotTheme

```

Wage tax losses since the beginning of 2020, relative to projections based on 2014-19, are still as much as $2 million per quarter by sector.

```{r projection_chart_bar, warning = FALSE, message = FALSE, echo=FALSE}
wage_tax_revenue_new %>%
  group_by(date, sector, year, month) %>% 
  summarize(Actual = sum(total), 
            Pre.Pandemic.Trend = sum(pred)) %>% 
  mutate(loss = Actual - Pre.Pandemic.Trend) %>%
  filter(year >= 2020) %>%
  ungroup() %>%
  select(date, sector, loss) %>%
  gather(-sector, - date, key = "variable", value = "value") %>%
  mutate(value_m = round(value/1000000, digits = 2)) %>%
  ggplot()+
    geom_bar(aes(x = as.Date(date), y=value_m), stat = "identity", color = "grey")+ 
  facet_wrap(~sector)+
    ylab("Quarterly Wage Tax Revenue shortfall ($ Millions)")+
    xlab("Date")+
    labs(
        title = "Philadelphia Wage Tax Receipts vs. Pre-Pandemic Trends",
        subtitle = "Adjusted for inflation to 2021 Dollars. Source: Philadelphia City Controller's Office, BLS")+
    plotTheme

```

Total tax shortfalls (compared to pre-pandemic trend) since March, 2020, in millions (2021 dollars)

```{r loss_summary, echo=FALSE, message=FALSE, warning=FALSE}

wage_tax_revenue_new %>%
  group_by(date, sector, year, month) %>% 
  summarize(Actual = sum(total), 
            Pre.Pandemic.Trend = sum(pred)) %>% 
  mutate(loss = Actual - Pre.Pandemic.Trend) %>%
  ungroup() %>%
  filter(year == 2020 & month >=3 | year > 2020) %>%
  group_by(sector) %>%
  summarize("Est Loss (millions)" = round(sum(loss)/1000000, digits = 2)) %>%
  kable()

```

# 2. Save Our Stages Grants

This repo contains interactive maps and analysis of SBA Grants given as part of the "Save Our Stages" Act passed by the US Congress.

The data being used here have been geocoded using the Google Geocoding service, which is highly accurate but not error free. Data are current as of August 17, 2021, and are available [from the SBA](https://data.sba.gov/dataset/svog/resource/33270c2a-f1c5-4dcb-bc98-aedcaec19ef3).

## 2.2. Grantee map

The following map shows the grantee locations.

```{r venues, echo = FALSE, message = FALSE, warning = FALSE}
geocoded_venues <- read.csv("~/GitHub/SaveOurStagesGrants/data/geocoded_venues.csv") %>%
  rename(acct_type = 'Account.Venue') %>%
  filter(City == "Philadelphia" & State == "PA")

l <- leaflet() %>% 
  addProviderTiles(providers$Esri.WorldTopoMap) %>%
  setView(lng = mean(geocoded_venues$lon, na.rm = TRUE),
          lat = mean(geocoded_venues$lat, na.rm = TRUE),
          zoom = 10) %>%
  addScaleBar(position = "topleft") %>%
  addCircleMarkers(data= geocoded_venues %>%
                     filter(acct_type == "Live performing arts organization operator"),
                   lng=~lon, 
                   lat=~lat,
                   radius =~ 1, 
                   fillOpacity =~ 0.5,
                   color= "blue", # "~pal(acct_type)"
                   label=~paste(svog_grantee, " | Grant $", svog_amount),
                   group= "Live performing arts organization operator") %>%
  addCircleMarkers(data= geocoded_venues %>%
                     filter(acct_type == "Live venue operator or promoter"),
                   lng=~lon, 
                   lat=~lat,
                   radius =~ 1, 
                   fillOpacity =~ 0.5,
                   color= "blue", # "~pal(acct_type)"
                   label=~paste(svog_grantee, " | Grant $", svog_amount),
                   group= "Live venue operator or promoter") %>%
  addCircleMarkers(data= geocoded_venues %>%
                     filter(acct_type == "Motion picture theater operator"),
                   lng=~lon, 
                   lat=~lat,
                   radius =~ 1, 
                   fillOpacity =~ 0.5,
                   color= "blue", # "~pal(acct_type)"
                   label=~paste(svog_grantee, " | Grant $", svog_amount),
                   group= "Motion picture theater operator") %>%
  addCircleMarkers(data= geocoded_venues %>%
                     filter(acct_type == "Museum Operator"),
                   lng=~lon, 
                   lat=~lat,
                   radius =~ 1, 
                   fillOpacity =~ 0.5,
                   color= "blue", # "~pal(acct_type)"
                   label=~paste(svog_grantee, " | Grant $", svog_amount),
                   group= "Museum Operator") %>%
  addCircleMarkers(data= geocoded_venues %>%
                     filter(acct_type == "Talent representative"),
                   lng=~lon, 
                   lat=~lat,
                   radius =~ 1, 
                   fillOpacity =~ 0.5,
                   color= "blue", # "~pal(acct_type)"
                   label=~paste(svog_grantee, " | Grant $", svog_amount),
                   group= "Talent representative") %>%
  addCircleMarkers(data= geocoded_venues %>%
                     filter(acct_type == "Theatrical producer"),
                   lng=~lon, 
                   lat=~lat,
                   radius =~ 1, 
                   fillOpacity =~ 0.5,
                   color= "blue", # "~pal(acct_type)"
                   label=~paste(svog_grantee, " | Grant $", svog_amount),
                   group= "Theatrical producer") %>%
  addLayersControl(
    overlayGroups = c("Live performing arts organization operator", 
                      "Live venue operator or promoter",
                      "Motion picture theater operator",
                      "Museum Operator",
                      "Talent representative",
                      "Theatrical producer") ,
    options = layersControlOptions(collapsed = TRUE)
    ) %>%
  hideGroup(c("Live performing arts organization operator", 
                      "Motion picture theater operator",
                      "Museum Operator",
                      "Talent representative",
                      "Theatrical producer"))

l

```
# 2.2. Philadelphia SVOG grant breakdown:

```{r svog_breakdown, message = FALSE, echo = FALSE, warning = FALSE}
geocoded_venues %>%
  summarize("Num Grants" = n(),
            "Grants Total" = sum(svog_amount),
            "Mean Grant" = mean(svog_amount),
            "Median Grant" = median(svog_amount)) %>%
  kable() %>%
  kable_styling()
```

## 2.3. Phila SVOG searchable database:

```{r searchable_table, echo=FALSE, message=FALSE, warning=FALSE}
datatable(geocoded_venues %>%
            select(-X, -lon, -lat, -address_full) %>%
            rename('Grant ($)' = svog_amount,
                   Grantee = svog_grantee,
                   Category = acct_type), 
          options = list(pageLength = 10))
```


# 3. Zoning and Assembly Licenses

A Special Assembly Occupancy License (SAOL) is the highest form of license for assembly for entertainment for 50+ people (non-seated). In combination with an amusement license and regular business licenses, this is what you usually need to open a venue. These licenses are difficult to get, and can be expensive. Much expense and risk is incurred when you need a zoning variance (and council approval) or to go through a zoning process triggered by a special exception.

SAOLs are only available by right in very few zoning categories, most of these the most expensive land in Center City. Recently, people have NOT been taking out licenses in CC, they have been going through zoning processes in nearby neighborhoods with transit access and clusters of amenities, and SAOLs have been going extinct much faster than they are being created.

There are a few opportunities for Council to make SAOLs easier to access - one is by supporting operators in getting variances, two is by opening up more zoning districts to SAOLs by right.

[24HrPHL put together a "playbook" for how to obtain a SAOL in 2019 that was vetted by then L&I Commissioner Dave Perri.](http://24hrphl.org/the-venue-starter-playbook/)

The number of active SAOLs in Philadelphia is declining, and the places where people want to (or can afford to) build creative spaces are not the places where they are available by-right. Only 28% of active SAOLs are in by-right districts. 

Frequently, people are applying for SAOLs in CMX-2.5 properties - where this use is availabe by special exception *Opening CMX-2.5 to by-right SAOL would more than double the amount of commercial parcels eligible for by-right SAOL.*

```{r dl_zoning, echo=FALSE, message=FALSE, warning=FALSE, echo=FALSE}

zones <- st_read("https://opendata.arcgis.com/datasets/0bdb0b5f13774c03abf8dc2f1aa01693_0.geojson") %>%
  st_as_sf(crs = 4326) %>%
  mutate(assembly_allowed = ifelse(LONG_CODE %in% 
                                     c("CMX-4", "CMX-5", "CA-2", "ICMX"),
                                   "By-Right", 
                                   ifelse(LONG_CODE %in% 
                                            c("CMX-2", "CMX-2.5", "CMX-3", "IRMX"),
                                          "Special Exception", 
                                          "Not Permitted" )))

licenses <- st_read("https://phl.carto.com/api/v2/sql?q=SELECT+*+FROM+business_licenses&filename=business_licenses&format=geojson&skipfields=cartodb_id") %>%
  st_as_sf(crs = 4326)

```
## 3.1. Summary of SAOL-eligible Districts

Assembly is allowed by right in four zoning districts, and by special exception in four others.

Here are some very basic descriptions of these types of zones, with information sourced from Anastasio Law - http://phillyzoning.com/

*By Right*

-CA-2 - Auto-oriented commercial

-CMX-4 - Commercial Mixed Use, mostly found in Center City or along major arterials like Broad, Market or Chestnut Streets

-CMX-5 - Commercial Mixed Use, found in the core of Center City office districts

-ICMX - Commercial/Industrial uses designed as a buffer between commercial and residential uses

*Special Exception*

-CMX-2 - Neighborhood commercial corridor, mixed use - e.g. Baltimore Avenue, Germantown Avenue

-CMX-2.5 - Commercial mixed use designed to promote pedestrian-friendly uses

-CMX-3 - Lower density commercial mixed use than CMX 4&5, found on major corridors like Kensington Ave, South St, Broad south of Washington.

-IRMX - "Low impact" industrial, including artist spaces

```{r explore_zones, message = FALSE, warning = FALSE, echo=FALSE, eval = FALSE}
zones %>%
  as.data.frame()%>%
  filter(LONG_CODE %in% c("CMX-4", "CMX-5", "CA-2", "ICMX", "CMX-2", "CMX-2.5", "CMX-3", "IRMX")) %>%
  group_by(assembly_allowed, LONG_CODE) %>%
  tally() %>%
  rename("Parcels city-wide" = n,
         District = LONG_CODE) %>%
  kable %>%
  kable_styling()
```

City-wide, roughly 26% of *commercial* districts (by area) are zoned for assembly by right. Much of this area is in large Center City tracts where creative spaces are unlikely to be developed due to cost or constraints of the land.

```{r pct_commercial_props, message = FALSE, warning = FALSE, echo=FALSE}
zones %>%
  filter(str_detect(ZONINGGROUP, "Commercial") == TRUE) %>%
  st_transform(2272) %>%
  mutate(area = st_area(.)) %>%
  as.data.frame()%>%
  group_by(assembly_allowed) %>%
  summarize(sum_area = sum(area)) %>%
  #rename(n_parcels = n) %>%
  mutate(Pct = round(100*(sum_area / sum(sum_area)), 2)) %>%
  #rename("Parcels City-Wide" = n_parcels) %>%
  kable() %>%
  kable_styling()
```

## 3.2. City-wide zoning examination

The majority of areas zoned for assembly by-right are in Center City. This is *not* where most development of creative entertainment is taking place.

Take a look at districts outside Center City like Fishtown - by-right SAOL is not really allowed by-right anywhere.

When the license data from OpenDataPhilly are back online - this map will contain SAOL licence locations as well.

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

```{r map_exceptions, message = FALSE, warning = FALSE, echo=FALSE, fig.width=8, fig.height= 8}
mapview(zones %>%
          filter(assembly_allowed %in% c("Special Exception", "By-Right")) %>%
          mutate(LONG_CODE = case_when(LONG_CODE == "CA-2" ~ "By-Right CA-2",
                                       LONG_CODE == "CMX-4" ~ "By-Right CMX-4",
                                       LONG_CODE == "CMX-5" ~ "By-Right CMX-5",
                                       LONG_CODE == "ICMX" ~ "By-Right ICMX",
                                       LONG_CODE == "CMX-2" ~ "Spec Execpt CMX-2",
                                       LONG_CODE == "CMX-2.5" ~ "Spec Execpt CMX-2.5",
                                       LONG_CODE == "CMX-3" ~ "Spec Execpt CMX-3",
                                       LONG_CODE == "IRMX" ~ "Spec Execpt IRMX")),
        zcol = "LONG_CODE", col.regions=list("#ffffb2", "#fecc5c", "#fd8d3c", "#e31a1c",
                                             "#edf8fb", "#b3cde3", "#8c96c6", "#88419d"),
        col = list("transparent"),
        layer.name = "Zoning Districts")


# +
#mapview(licenses %>% 
#  filter(licensetype == "Special Assembly Occupancy",
#         licensestatus == "Active") %>% 
#  st_join(., zones), zcol = "assembly_allowed")  
```


## 3.4. SAOL issuance timeline

Very few SAOLs are issued in Philly, and especially few the last two years. The following chart shows the date when SAOL's were issued. 2006 probably has data inflated by the start of the record-keeping period for these data.

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

```{r license_timeline, eval=FALSE, message=FALSE, warning=FALSE, include=FALSE}
licenses %>%
  filter(licensetype == "Special Assembly Occupancy") %>%
  mutate(year_opened = year(as.Date(initialissuedate))) %>%
  group_by(year_opened) %>%
  tally() %>%
  ggplot()+
  geom_histogram(aes(y = n, x = year_opened), stat = "identity") +
  labs(title="Special Assembly Occupancy License Active Dates",
       x="Year", 
       y="Number of Licenses")+
  plotTheme

```

Record numbers of licenses became inactive in 2020... The following chart shows the years in which SAOL's became inactive. Meanwhile, the real estate market is getting much more competitive and entertainment wages are slow to recover (see Section 1 of this document).

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

```{r license_timeline_inactive, eval=FALSE, message=FALSE, warning=FALSE, include=FALSE}
licenses %>%
  filter(licensetype == "Special Assembly Occupancy",
         is.na(inactivedate) == FALSE) %>%
  mutate(year_closed = year(as.Date(inactivedate))) %>%
  group_by(year_closed) %>%
  tally() %>%
  ggplot()+
  geom_histogram(aes(y = n, x = year_closed), stat = "identity") +
  labs(title="Special Assembly Occupancy License Inactive Dates",
       x="Year", 
       y="Number of Licenses")+
  plotTheme

```

```{r map_licenses, echo=FALSE, eval = FALSE, include = FALSE}
mapview(licenses %>% 
  filter(licensetype == "Special Assembly Occupancy",
         licensestatus == "Active") %>% 
  st_join(., zones), zcol = "assembly_allowed")
```

Among active Special Assembly Occupancy Licenses, only 35 (28%) were permitted by right in the zones where they were created.

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

```{r licenses_and_zones, echo=FALSE, eval = FALSE, include = FALSE}
licenses %>% 
  filter(licensetype == "Special Assembly Occupancy",
         licensestatus == "Active") %>% 
  st_join(., zones) %>% 
  as.data.frame() %>%
  group_by(assembly_allowed, LONG_CODE) %>% 
  tally() %>%
  rename(n_licenses = n) %>%
  left_join(., zones %>%
              as.data.frame() %>%
              group_by(LONG_CODE) %>%
              tally() %>%
              rename( total_zone_parcels = n)) %>%
  group_by(assembly_allowed) %>%
  summarize(n_licenses = sum(n_licenses)) %>%
  mutate(pct = 100*(n_licenses / sum(n_licenses))) %>%
  kable()
  
```

Next steps - 

How has by-right vs. exception/variance changed over time?
How is this related to property costs / tax rates / valuation?
How many SAOLs are also getting amusement licenses? How many can we verify are music venues?
Are temporary assembly licenses in the data set?

DATA FROM OPENDATAPHILLY UNAVAILABLE 5/8/2022

# Next up for analysis

overlays, Mobility trends, PPP data, Illuminate the Arts grants


Bike Share and Mobility

Philadelphia Bike Share Q2, 2021

```{r load_mobility, message=TRUE, warning=FALSE, include=FALSE, eval = FALSE}
url <-"https://u626n26h74f16ig1p3pt0f2g-wpengine.netdna-ssl.com/wp-content/uploads/2021/04/indego-trips-2021-q1.zip"

temp <- tempfile()
temp2 <- tempfile()

download.file(url, temp)
unzip(zipfile = temp, exdir = temp2)
q1_2021 <- read.csv(file.path(temp2, "indego-trips-2021-q1.csv"))

unlink(c(temp, temp2))

url <-"https://u626n26h74f16ig1p3pt0f2g-wpengine.netdna-ssl.com/wp-content/uploads/2021/07/indego-trips-2021-q2.zip"

temp <- tempfile()
temp2 <- tempfile()

download.file(url, temp)
unzip(zipfile = temp, exdir = temp2)
q2_2021 <- read.csv(file.path(temp2, "indego-trips-2021-q2.csv"))

unlink(c(temp, temp2))

url <-"https://u626n26h74f16ig1p3pt0f2g-wpengine.netdna-ssl.com/wp-content/uploads/2021/10/indego-trips-2021-q3.zip"

temp <- tempfile()
temp2 <- tempfile()

download.file(url, temp)
unzip(zipfile = temp, exdir = temp2)
q3_2021 <- read.csv(file.path(temp2, "indego-trips-2021-q3.csv"))

unlink(c(temp, temp2))

url <-"https://u626n26h74f16ig1p3pt0f2g-wpengine.netdna-ssl.com/wp-content/uploads/2022/01/indego-trips-2021-q4.zip"

temp <- tempfile()
temp2 <- tempfile()

download.file(url, temp)
unzip(zipfile = temp, exdir = temp2)
q4_2021 <- read.csv(file.path(temp2, "indego-trips-2021-q4.csv"))

unlink(c(temp, temp2))

dat <- q1_2021 %>%
  rbind(., q2_2021) %>%
  rbind(., q3_2021) %>%
  rbind(., q4_2021)

remove(q1_2021)
remove(q2_2021)
remove(q3_2021)
remove(q4_2021)

phila_shp <- counties("PA") %>%
  filter(NAME == "Philadelphia") %>%
  st_as_sf(crs = 4326)

dat2 <- dat %>%
  mutate(interval60 = floor_date(mdy_hm(start_time), unit = "hour"),
         interval15 = floor_date(mdy_hm(start_time), unit = "15 mins"),
         week = week(interval60),
         dotw = wday(interval60, label=TRUE))
```


```{r mobility_trends, warning = FALSE, echo = TRUE, message = FALSE, eval = FALSE}
ggplot(dat2 %>%
         group_by(week) %>%
         tally())+
  geom_line(aes(x = week, y = n))+
  labs(title="Bike share trips per week Philadelphia, 2021",
       x="Week (2021)", 
       y="Trips")+
  plotTheme

ggplot(dat2 %>% mutate(hour = hour(mdy_hm(start_time)),
                       month = month(mdy_hm(start_time)),
                       day_type = ifelse(dotw %in% c("Sat", "Sun", "Fri"), "Weekend", "Weekday"),
                       month = case_when(month == 1 ~ "01-2021",
                                         month == 2 ~ "02-2021",
                                         month == 3 ~ "03-2021",
                                         month == 4 ~ "04-2021",
                                         month == 5 ~ "05-2021",
                                         month == 6 ~ "06-2021",
                                         month == 7 ~ "07-2021",
                                         month == 8 ~ "08-2021",
                                         month == 9 ~ "09-2021",
                                         month == 10 ~ "10-2021",
                                         month == 11 ~ "11-2021",
                                         month == 12 ~ "12-2021")))+
  geom_rect(aes(xmin=0,
                xmax = 6,
                ymin = 0,
                ymax = Inf), fill = 'light grey', alpha = 0.05)+
  geom_rect(aes(xmin=18,
                xmax = 24,
                ymin = 0,
                ymax = Inf), fill = 'light grey', alpha = 0.05)+
  geom_freqpoly(aes(hour, color = day_type), binwidth = 1)+
  scale_x_continuous(name="Hour", breaks=seq(0,24,6))+
  labs(title="Total Bike Share Trips",
       subtitle = "Philadelphia, 2021",
       x="Hour", 
       y="Trips")+
  facet_wrap(~month)+
  theme(legend.title = element_blank())+
  plotTheme
```



```{r mobility_map, eval=FALSE, message=FALSE, warning=FALSE, include=FALSE}
ggplot(dat2 %>%
         mutate(time_of_day = ifelse(hour(interval60) < 4 | 
                                       hour(interval60) > 20, 
                                     "Night", 
                                     "Day"),
                month = month(mdy_hm(start_time)),
                month = case_when(month == 1 ~ "01-2021",
                                                    month == 2 ~ "02-2021",
                                                    month == 3 ~ "03-2021",
                                                    month == 4 ~ "04-2021",
                                                    month == 5 ~ "05-2021",
                                                    month == 6 ~ "06-2021",
                                                    month == 7 ~ "07-2021",
                                                    month == 8 ~ "08-2021",
                                                    month == 9 ~ "09-2021",
                                                    month == 10 ~ "10-2021",
                                                    month == 11 ~ "11-2021",
                                                    month == 12 ~ "12-2021")) %>%
         group_by(start_station, time_of_day, start_lon, start_lat, month) %>%
         tally() %>%
         filter(time_of_day == "Night")) +
  geom_sf(data = phila_shp, fill = "black")+
  geom_point(aes(x = start_lon, y = start_lat, color = n), alpha = 0.8)+
  scale_color_viridis_c()+
  guides(color=guide_legend(title="Trips"))+
  labs(title="Gross Bike Share Trips By Origin - 20:00-4:00, Q2, 2021")+
  facet_wrap(~month)+
  mapTheme
```